Spring ContextLoaderListener 启动流程解析一
本篇博客主要讲的 spring 如何初始化其上下文的源码简单解读。
web.xml
web.xml 即应用程序部署描述符文件,其中包含了应用程序需要初始化的组件,比如 Filter、Listener、Servle、欢迎页、session配置等。
- web.xml tomcat官网介绍 https://tomcat.apache.org/tomcat-7.0-doc/appdev/deployment.html
- web.xml 介绍 https://blog.csdn.net/ckc_666/article/details/82964812
ContextLoaderListener
<context-param>标签,用于配置一个全局变量,spring ContextLoaderListener 会使用到这个全局变量,
<context-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:spring.xml</param-value>
</context-param>
<listener>
<description>spring监听器</description>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
ContextLoaderListener 实现了 ServletContextListener 接口。ServletContextListener 接口够监听 ServletContext 对象的生命周期,实际上就是监听 Web 应用的生命周期。当 Servlet 容器启动或终止 Web 应用时,会触发 ServletContextEvent 事件,该事件由 ServletContextListener 来处理。在web容器启动时会触发 ContextLoaderListener 中的 contextInitialized 方法。
ContextLoaderListener 代码如下
public class ContextLoaderListener extends ContextLoader implements ServletContextListener {
public ContextLoaderListener() {
}
public ContextLoaderListener(WebApplicationContext context) {
super(context);
}
public void contextInitialized(ServletContextEvent event) {
//调用这个方法,看名字就知道是初始化 spring 的上下文
this.initWebApplicationContext(event.getServletContext());
}
public void contextDestroyed(ServletContextEvent event) {
this.closeWebApplicationContext(event.getServletContext());
ContextCleanupListener.cleanupAttributes(event.getServletContext());
}
}
ContextLoader.initWebApplicationContext
public WebApplicationContext initWebApplicationContext(ServletContext servletContext) {
// 先判断 ServletContext 中是否已存在上下文,存在说明已加载或配置信息有误
if (servletContext.getAttribute(WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE) != null) {
throw new IllegalStateException("Cannot initialize context because there is already a root application context present - check whether you have multiple ContextLoader* definitions in your web.xml!");
} else {
...
try {
if (this.context == null) {
//创建 WebApplicationContext
this.context = this.createWebApplicationContext(servletContext);
}
if (this.context instanceof ConfigurableWebApplicationContext) {
//如果 this.context 是 ConfigurableWebApplicationContext 的实例,则配置并刷新 WebApplicationContext
ConfigurableWebApplicationContext cwac = (ConfigurableWebApplicationContext)this.context;
if (!cwac.isActive()) {
if (cwac.getParent() == null) {
ApplicationContext parent = this.loadParentContext(servletContext);
cwac.setParent(parent);
}
this.configureAndRefreshWebApplicationContext(cwac, servletContext);
}
}
//将 WebApplicationContext 放入 servletContext 中,供后续流程使用
servletContext.setAttribute(WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE, this.context);
ClassLoader ccl = Thread.currentThread().getContextClassLoader();
if (ccl == ContextLoader.class.getClassLoader()) {
currentContext = this.context;
} else if (ccl != null) {
currentContextPerThread.put(ccl, this.context);
}
...
return this.context;
} catch (RuntimeException var8) {
logger.error("Context initialization failed", var8);
servletContext.setAttribute(WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE, var8);
throw var8;
} catch (Error var9) {
logger.error("Context initialization failed", var9);
servletContext.setAttribute(WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE, var9);
throw var9;
}
}
}
ContextLoader.configureAndRefreshWebApplicationContext
配置和刷新上下文,主要是设置容器 id 和获得配置文件路径
protected void configureAndRefreshWebApplicationContext(ConfigurableWebApplicationContext wac, ServletContext sc) {
String configLocationParam;
//检查并设置容器 id
if (ObjectUtils.identityToString(wac).equals(wac.getId())) {
configLocationParam = sc.getInitParameter("contextId");
//如果配置文件配置了容器 id,使用配置的容器 id
if (configLocationParam != null) {
wac.setId(configLocationParam);
} else {
wac.setId(ConfigurableWebApplicationContext.APPLICATION_CONTEXT_ID_PREFIX + ObjectUtils.getDisplayString(sc.getContextPath()));
}
}
wac.setServletContext(sc);
//获取 spring 配置文件,我们在 web.xml 里有配置
configLocationParam = sc.getInitParameter("contextConfigLocation");
if (configLocationParam != null) {
wac.setConfigLocation(configLocationParam);
}
ConfigurableEnvironment env = wac.getEnvironment();
if (env instanceof ConfigurableWebEnvironment) {
((ConfigurableWebEnvironment)env).initPropertySources(sc, (ServletConfig)null);
}
//用户自定义上下文,这里暂时不讲
this.customizeContext(sc, wac);
//刷新上下文,读取配置文件,解析bean定义,注册BeanDefinition,实例化bean,完成注入的过程
wac.refresh();
}
AbstractApplicationContext.refresh()
refresh() 方法的实现类是抽象类 AbstractApplicationContext,继承了 ConfigurableApplicationContext 等接口。
refresh() 做了所有的上下文初始化动作,包括获取 BeanFactory,处理 BeanFactory,实例化 bean 等。
public void refresh() throws BeansException, IllegalStateException {
synchronized(this.startupShutdownMonitor) {
//容器刷新前准备工作
this.prepareRefresh();
//启动 obtainFreshBeanFactory 里的 refreshBeanFactory() 方法
//主要是把旧的 beanFactory 关闭,重新创建一个 beanFactory
ConfigurableListableBeanFactory beanFactory = this.obtainFreshBeanFactory();
this.prepareBeanFactory(beanFactory);
try {
this.postProcessBeanFactory(beanFactory);
this.invokeBeanFactoryPostProcessors(beanFactory);
this.registerBeanPostProcessors(beanFactory);
this.initMessageSource();
this.initApplicationEventMulticaster();
this.onRefresh();
this.registerListeners();
this.finishBeanFactoryInitialization(beanFactory);
this.finishRefresh();
} catch (BeansException var9) {
if (this.logger.isWarnEnabled()) {
this.logger.warn("Exception encountered during context initialization - cancelling refresh attempt: " + var9);
}
this.destroyBeans();
this.cancelRefresh(var9);
throw var9;
} finally {
this.resetCommonCaches();
}
}
}
接下来
接下来我们讲 beanFactory 的获取。<>
转载请注明来源,欢迎对文章中的引用来源进行考证,欢迎指出任何有错误或不够清晰的表达。可以在下面评论区评论,也可以邮件至 rockeycui@163.com
文章标题:Spring ContextLoaderListener 启动流程解析一
文章字数:889
本文作者:崔石磊(RockeyCui)
发布时间:2018-09-22, 21:00:00
原始链接:https://cuishilei.com/Spring ContextLoaderListener 启动流程解析一.html版权声明: "署名-非商用-相同方式共享 4.0" 转载请保留原文链接及作者。